home *** CD-ROM | disk | FTP | other *** search
/ Language/OS - Multiplatform Resource Library / LANGUAGE OS.iso / glass / glass.lha / GLASS / contsens / p2clib.c < prev    next >
C/C++ Source or Header  |  1991-01-31  |  17KB  |  950 lines

  1.  
  2. /* Run-time library for use with "p2c", the Pascal to C translator */
  3.  
  4. /* "p2c"  Copyright (C) 1989 Dave Gillespie.
  5.  * This file may be copied, modified, etc. in any way.  It is not restricted
  6.  * by the licence agreement accompanying p2c itself.
  7.  */
  8.  
  9.  
  10.  
  11. #include "p2c.h"
  12.  
  13.  
  14. /* #define LACK_LABS     */   /* Define these if necessary */
  15. /* #define LACK_MEMMOVE  */
  16.  
  17.  
  18. #ifndef NO_TIME
  19. # include <time.h>
  20. #endif
  21.  
  22.  
  23. #define Isspace(c)  isspace(c)      /* or "((c) == ' ')" if preferred */
  24.  
  25.  
  26.  
  27.  
  28. int P_argc;
  29. char **P_argv;
  30.  
  31. short P_escapecode;
  32. int P_ioresult;
  33.  
  34. long EXCP_LINE;    /* Used by Pascal workstation system */
  35.  
  36. Anyptr __MallocTemp__;
  37.  
  38. __p2c_jmp_buf *__top_jb;
  39.  
  40.  
  41.  
  42.  
  43. void PASCAL_MAIN(argc, argv)
  44. int argc;
  45. char **argv;
  46. {
  47.     P_argc = argc;
  48.     P_argv = argv;
  49.     __top_jb = NULL;
  50.  
  51. #ifdef LOCAL_INIT
  52.     LOCAL_INIT();
  53. #endif
  54. }
  55.  
  56.  
  57.  
  58.  
  59.  
  60. /* In case your system lacks these... */
  61.  
  62. #ifdef LACK_LABS
  63. long labs(x)
  64. long x;
  65. {
  66.     return((x > 0) ? x : -x);
  67. }
  68. #endif
  69.  
  70.  
  71. #ifdef LACK_MEMMOVE
  72. Anyptr memmove(d, s, n)
  73. Anyptr d, s;
  74. register long n;
  75. {
  76.     if (d < s || d - s >= n) {
  77.     memcpy(d, s, n);
  78.     return d;
  79.     } else if (n > 0) {
  80.     register char *dd = d + n, *ss = s + n;
  81.     while (--n >= 0)
  82.         *--dd = *--ss;
  83.     }
  84.     return d;
  85. }
  86. #endif
  87.  
  88.  
  89. int my_toupper(c)
  90. int c;
  91. {
  92.     if (islower(c))
  93.     return _toupper(c);
  94.     else
  95.     return c;
  96. }
  97.  
  98.  
  99. int my_tolower(c)
  100. int c;
  101. {
  102.     if (isupper(c))
  103.     return _tolower(c);
  104.     else
  105.     return c;
  106. }
  107.  
  108.  
  109.  
  110.  
  111. long ipow(a, b)
  112. long a, b;
  113. {
  114.     long v;
  115.  
  116.     if (a == 0 || a == 1)
  117.     return a;
  118.     if (a == -1)
  119.     return (b & 1) ? -1 : 1;
  120.     if (b < 0)
  121.     return 0;
  122.     if (a == 2)
  123.     return 1 << b;
  124.     v = (b & 1) ? a : 1;
  125.     while ((b >>= 1) > 0) {
  126.     a *= a;
  127.     if (b & 1)
  128.         v *= a;
  129.     }
  130.     return v;
  131. }
  132.  
  133.  
  134.  
  135.  
  136. /* Common string functions: */
  137.  
  138. /* Store in "ret" the substring of length "len" starting from "pos" (1-based).
  139.    Store a shorter or null string if out-of-range.  Return "ret". */
  140.  
  141. char *strsub(ret, s, pos, len)
  142. register char *ret, *s;
  143. register int pos, len;
  144. {
  145.     register char *s2;
  146.  
  147.     if (--pos < 0 || len <= 0) {
  148.         *ret = 0;
  149.         return ret;
  150.     }
  151.     while (pos > 0) {
  152.         if (!*s++) {
  153.             *ret = 0;
  154.             return ret;
  155.         }
  156.         pos--;
  157.     }
  158.     s2 = ret;
  159.     while (--len >= 0) {
  160.         if (!(*s2++ = *s++))
  161.             return ret;
  162.     }
  163.     *s2 = 0;
  164.     return ret;
  165. }
  166.  
  167.  
  168. /* Return the index of the first occurrence of "pat" as a substring of "s",
  169.    starting at index "pos" (1-based).  Result is 1-based, 0 if not found. */
  170.  
  171. int strpos2(s, pat, pos)
  172. char *s;
  173. register char *pat;
  174. register int pos;
  175. {
  176.     register char *cp, ch;
  177.     register int slen;
  178.  
  179.     if (--pos < 0)
  180.         return 0;
  181.     slen = strlen(s) - pos;
  182.     cp = s + pos;
  183.     if (!(ch = *pat++))
  184.         return 0;
  185.     pos = strlen(pat);
  186.     slen -= pos;
  187.     while (--slen >= 0) {
  188.         if (*cp++ == ch && !strncmp(cp, pat, pos))
  189.             return cp - s;
  190.     }
  191.     return 0;
  192. }
  193.  
  194.  
  195. /* Case-insensitive version of strcmp. */
  196.  
  197. int strcicmp(s1, s2)
  198. register char *s1, *s2;
  199. {
  200.     register unsigned char c1, c2;
  201.  
  202.     while (*s1) {
  203.     if (*s1++ != *s2++) {
  204.         if (!s2[-1])
  205.         return 1;
  206.         c1 = toupper(s1[-1]);
  207.         c2 = toupper(s2[-1]);
  208.         if (c1 != c2)
  209.         return c1 - c2;
  210.     }
  211.     }
  212.     if (*s2)
  213.     return -1;
  214.     return 0;
  215. }
  216.  
  217.  
  218.  
  219.  
  220. /* HP and Turbo Pascal string functions: */
  221.  
  222. /* Trim blanks at left end of string. */
  223.  
  224. char *strltrim(s)
  225. register char *s;
  226. {
  227.     while (Isspace(*s++)) ;
  228.     return s - 1;
  229. }
  230.  
  231.  
  232. /* Trim blanks at right end of string. */
  233.  
  234. char *strrtrim(s)
  235. register char *s;
  236. {
  237.     register char *s2 = s;
  238.  
  239.     if (!*s)
  240.     return s;
  241.     while (*++s2) ;
  242.     while (s2 > s && Isspace(*--s2))
  243.         *s2 = 0;
  244.     return s;
  245. }
  246.  
  247.  
  248. /* Store in "ret" "num" copies of string "s".  Return "ret". */
  249.  
  250. char *strrpt(ret, s, num)
  251. char *ret;
  252. register char *s;
  253. register int num;
  254. {
  255.     register char *s2 = ret;
  256.     register char *s1;
  257.  
  258.     while (--num >= 0) {
  259.         s1 = s;
  260.         while ((*s2++ = *s1++)) ;
  261.         s2--;
  262.     }
  263.     return ret;
  264. }
  265.  
  266.  
  267. /* Store in "ret" string "s" with enough pad chars added to reach "size". */
  268.  
  269. char *strpad(ret, s, padchar, num)
  270. char *ret;
  271. register char *s;
  272. register int padchar, num;
  273. {
  274.     register char *d = ret;
  275.  
  276.     if (s == d) {
  277.     while (*d++) ;
  278.     } else {
  279.     while ((*d++ = *s++)) ;
  280.     }
  281.     num -= (--d - ret);
  282.     while (--num >= 0)
  283.     *d++ = padchar;
  284.     *d = 0;
  285.     return ret;
  286. }
  287.  
  288.  
  289. /* Copy the substring of length "len" from index "spos" of "s" (1-based)
  290.    to index "dpos" of "d", lengthening "d" if necessary.  Length and
  291.    indices must be in-range. */
  292.  
  293. void strmove(len, s, spos, d, dpos)
  294. register char *s, *d;
  295. register int len, spos, dpos;
  296. {
  297.     s += spos - 1;
  298.     d += dpos - 1;
  299.     while (*d && --len >= 0)
  300.     *d++ = *s++;
  301.     if (len > 0) {
  302.     while (--len >= 0)
  303.         *d++ = *s++;
  304.     *d = 0;
  305.     }
  306. }
  307.  
  308.  
  309. /* Delete the substring of length "len" at index "pos" from "s".
  310.    Delete less if out-of-range. */
  311.  
  312. void strdelete(s, pos, len)
  313. register char *s;
  314. register int pos, len;
  315. {
  316.     register int slen;
  317.  
  318.     if (--pos < 0)
  319.         return;
  320.     slen = strlen(s) - pos;
  321.     if (slen <= 0)
  322.         return;
  323.     s += pos;
  324.     if (slen <= len) {
  325.         *s = 0;
  326.         return;
  327.     }
  328.     while ((*s = s[len])) s++;
  329. }
  330.  
  331.  
  332. /* Insert string "src" at index "pos" of "dst". */
  333.  
  334. void strinsert(src, dst, pos)
  335. register char *src, *dst;
  336. register int pos;
  337. {
  338.     register int slen, dlen;
  339.  
  340.     if (--pos < 0)
  341.         return;
  342.     dlen = strlen(dst);
  343.     dst += dlen;
  344.     dlen -= pos;
  345.     if (dlen <= 0) {
  346.         strcpy(dst, src);
  347.         return;
  348.     }
  349.     slen = strlen(src);
  350.     do {
  351.         dst[slen] = *dst;
  352.         --dst;
  353.     } while (--dlen >= 0);
  354.     dst++;
  355.     while (--slen >= 0)
  356.         *dst++ = *src++;
  357. }
  358.  
  359.  
  360.  
  361.  
  362. /* File functions */
  363.  
  364. /* Peek at next character of input stream; return EOF at end-of-file. */
  365.  
  366. int P_peek(f)
  367. FILE *f;
  368. {
  369.     int ch;
  370.  
  371.     ch = getc(f);
  372.     if (ch == EOF)
  373.     return EOF;
  374.     ungetc(ch, f);
  375.     return (ch == '\n') ? ' ' : ch;
  376. }
  377.  
  378.  
  379. /* Check if at end of file, using Pascal "eof" semantics.  End-of-file for
  380.    stdin is broken; remove the special case for it to be broken in a
  381.    different way. */
  382.  
  383. int P_eof(f)
  384. FILE *f;
  385. {
  386.     register int ch;
  387.  
  388.     if (feof(f))
  389.     return 1;
  390.     if (f == stdin)
  391.     return 0;    /* not safe to look-ahead on the keyboard! */
  392.     ch = getc(f);
  393.     if (ch == EOF)
  394.     return 1;
  395.     ungetc(ch, f);
  396.     return 0;
  397. }
  398.  
  399.  
  400. /* Check if at end of line (or end of entire file). */
  401.  
  402. int P_eoln(f)
  403. FILE *f;
  404. {
  405.     register int ch;
  406.  
  407.     ch = getc(f);
  408.     if (ch == EOF)
  409.         return 1;
  410.     ungetc(ch, f);
  411.     return (ch == '\n');
  412. }
  413.  
  414.  
  415. /* Read a packed array of characters from a file. */
  416.  
  417. Void P_readpaoc(f, s, len)
  418. FILE *f;
  419. char *s;
  420. int len;
  421. {
  422.     int ch;
  423.  
  424.     for (;;) {
  425.     if (len <= 0)
  426.         return;
  427.     ch = getc(f);
  428.     if (ch == EOF || ch == '\n')
  429.         break;
  430.     *s++ = ch;
  431.     --len;
  432.     }
  433.     while (--len >= 0)
  434.     *s++ = ' ';
  435.     if (ch != EOF)
  436.     ungetc(ch, f);
  437. }
  438.  
  439. Void P_readlnpaoc(f, s, len)
  440. FILE *f;
  441. char *s;
  442. int len;
  443. {
  444.     int ch;
  445.  
  446.     for (;;) {
  447.     ch = getc(f);
  448.     if (ch == EOF || ch == '\n')
  449.         break;
  450.     if (len > 0) {
  451.         *s++ = ch;
  452.         --len;
  453.     }
  454.     }
  455.     while (--len >= 0)
  456.     *s++ = ' ';
  457. }
  458.  
  459.  
  460. /* Compute maximum legal "seek" index in file (0-based). */
  461.  
  462. long P_maxpos(f)
  463. FILE *f;
  464. {
  465.     long savepos = ftell(f);
  466.     long val;
  467.  
  468.     if (fseek(f, 0L, SEEK_END))
  469.         return -1;
  470.     val = ftell(f);
  471.     if (fseek(f, savepos, SEEK_SET))
  472.         return -1;
  473.     return val;
  474. }
  475.  
  476.  
  477. /* Use packed array of char for a file name. */
  478.  
  479. Char *P_trimname(fn, len)
  480. register Char *fn;
  481. register int len;
  482. {
  483.     static Char fnbuf[256];
  484.     register Char *cp = fnbuf;
  485.     
  486.     while (--len >= 0 && *fn && !isspace(*fn))
  487.     *cp++ = *fn++;
  488.     return fnbuf;
  489. }
  490.  
  491.  
  492.  
  493.  
  494. /* Pascal's "memavail" doesn't make much sense in Unix with virtual memory.
  495.    We fix memory size as 10Meg as a reasonable compromise. */
  496.  
  497. long memavail()
  498. {
  499.     return 10000000;            /* worry about this later! */
  500. }
  501.  
  502. long maxavail()
  503. {
  504.     return memavail();
  505. }
  506.  
  507.  
  508.  
  509.  
  510. /* Sets are stored as an array of longs.  S[0] is the size of the set;
  511.    S[N] is the N'th 32-bit chunk of the set.  S[0] equals the maximum
  512.    I such that S[I] is nonzero.  S[0] is zero for an empty set.  Within
  513.    each long, bits are packed from lsb to msb.  The first bit of the
  514.    set is the element with ordinal value 0.  (Thus, for a "set of 5..99",
  515.    the lowest five bits of the first long are unused and always zero.) */
  516.  
  517. /* (Sets with 32 or fewer elements are normally stored as plain longs.) */
  518.  
  519. long *P_setunion(d, s1, s2)         /* d := s1 + s2 */
  520. register long *d, *s1, *s2;
  521. {
  522.     long *dbase = d++;
  523.     register int sz1 = *s1++, sz2 = *s2++;
  524.     while (sz1 > 0 && sz2 > 0) {
  525.         *d++ = *s1++ | *s2++;
  526.     sz1--, sz2--;
  527.     }
  528.     while (--sz1 >= 0)
  529.     *d++ = *s1++;
  530.     while (--sz2 >= 0)
  531.     *d++ = *s2++;
  532.     *dbase = d - dbase - 1;
  533.     return dbase;
  534. }
  535.  
  536.  
  537. long *P_setint(d, s1, s2)           /* d := s1 * s2 */
  538. register long *d, *s1, *s2;
  539. {
  540.     long *dbase = d++;
  541.     register int sz1 = *s1++, sz2 = *s2++;
  542.     while (--sz1 >= 0 && --sz2 >= 0)
  543.         *d++ = *s1++ & *s2++;
  544.     while (--d > dbase && !*d) ;
  545.     *dbase = d - dbase;
  546.     return dbase;
  547. }
  548.  
  549.  
  550. long *P_setdiff(d, s1, s2)          /* d := s1 - s2 */
  551. register long *d, *s1, *s2;
  552. {
  553.     long *dbase = d++;
  554.     register int sz1 = *s1++, sz2 = *s2++;
  555.     while (--sz1 >= 0 && --sz2 >= 0)
  556.         *d++ = *s1++ & ~*s2++;
  557.     if (sz1 >= 0) {
  558.         while (sz1-- >= 0)
  559.             *d++ = *s1++;
  560.     }
  561.     while (--d > dbase && !*d) ;
  562.     *dbase = d - dbase;
  563.     return dbase;
  564. }
  565.  
  566.  
  567. long *P_setxor(d, s1, s2)         /* d := s1 / s2 */
  568. register long *d, *s1, *s2;
  569. {
  570.     long *dbase = d++;
  571.     register int sz1 = *s1++, sz2 = *s2++;
  572.     while (sz1 > 0 && sz2 > 0) {
  573.         *d++ = *s1++ ^ *s2++;
  574.     sz1--, sz2--;
  575.     }
  576.     while (--sz1 >= 0)
  577.     *d++ = *s1++;
  578.     while (--sz2 >= 0)
  579.     *d++ = *s2++;
  580.     while (--d > dbase && !*d) ;
  581.     *dbase = d - dbase;
  582.     return dbase;
  583. }
  584.  
  585.  
  586. int P_inset(val, s)                 /* val IN s */
  587. register unsigned val;
  588. register long *s;
  589. {
  590.     register int bit;
  591.     bit = val % SETBITS;
  592.     val /= SETBITS;
  593.     if (val < *s++ && ((1<<bit) & s[val]))
  594.     return 1;
  595.     return 0;
  596. }
  597.  
  598.  
  599. long *P_addset(s, val)              /* s := s + [val] */
  600. register long *s;
  601. register unsigned val;
  602. {
  603.     register long *sbase = s;
  604.     register int bit, size;
  605.     bit = val % SETBITS;
  606.     val /= SETBITS;
  607.     size = *s;
  608.     if (++val > size) {
  609.         s += size;
  610.         while (val > size)
  611.             *++s = 0, size++;
  612.         *sbase = size;
  613.     } else
  614.         s += val;
  615.     *s |= 1<<bit;
  616.     return sbase;
  617. }
  618.  
  619.  
  620. long *P_addsetr(s, v1, v2)              /* s := s + [v1..v2] */
  621. register long *s;
  622. register unsigned v1, v2;
  623. {
  624.     register long *sbase = s;
  625.     register int b1, b2, size;
  626.     if ((int)v1 > (int)v2)
  627.     return sbase;
  628.     b1 = v1 % SETBITS;
  629.     v1 /= SETBITS;
  630.     b2 = v2 % SETBITS;
  631.     v2 /= SETBITS;
  632.     size = *s;
  633.     v1++;
  634.     if (++v2 > size) {
  635.         while (v2 > size)
  636.             s[++size] = 0;
  637.         s[v2] = 0;
  638.         *s = v2;
  639.     }
  640.     s += v1;
  641.     if (v1 == v2) {
  642.         *s |= (~((-2)<<(b2-b1))) << b1;
  643.     } else {
  644.         *s++ |= (-1) << b1;
  645.         while (++v1 < v2)
  646.             *s++ = -1;
  647.         *s |= ~((-2) << b2);
  648.     }
  649.     return sbase;
  650. }
  651.  
  652.  
  653. long *P_remset(s, val)              /* s := s - [val] */
  654. register long *s;
  655. register unsigned val;
  656. {
  657.     register int bit;
  658.     bit = val % SETBITS;
  659.     val /= SETBITS;
  660.     if (++val <= *s) {
  661.     if (!(s[val] &= ~(1<<bit)))
  662.         while (*s && !s[*s])
  663.         (*s)--;
  664.     }
  665.     return s;
  666. }
  667.  
  668.  
  669. int P_setequal(s1, s2)              /* s1 = s2 */
  670. register long *s1, *s2;
  671. {
  672.     register int size = *s1++;
  673.     if (*s2++ != size)
  674.         return 0;
  675.     while (--size >= 0) {
  676.         if (*s1++ != *s2++)
  677.             return 0;
  678.     }
  679.     return 1;
  680. }
  681.  
  682.  
  683. int P_subset(s1, s2)                /* s1 <= s2 */
  684. register long *s1, *s2;
  685. {
  686.     register int sz1 = *s1++, sz2 = *s2++;
  687.     if (sz1 > sz2)
  688.         return 0;
  689.     while (--sz1 >= 0) {
  690.         if (*s1++ & ~*s2++)
  691.             return 0;
  692.     }
  693.     return 1;
  694. }
  695.  
  696.  
  697. long *P_setcpy(d, s)                /* d := s */
  698. register long *d, *s;
  699. {
  700.     register long *save_d = d;
  701.  
  702. #ifdef SETCPY_MEMCPY
  703.     memcpy(d, s, (*s + 1) * sizeof(long));
  704. #else
  705.     register int i = *s + 1;
  706.     while (--i >= 0)
  707.         *d++ = *s++;
  708. #endif
  709.     return save_d;
  710. }
  711.  
  712.  
  713. /* s is a "smallset", i.e., a 32-bit or less set stored
  714.    directly in a long. */
  715.  
  716. long *P_expset(d, s)                /* d := s */
  717. register long *d;
  718. register long s;
  719. {
  720.     if (s) {
  721.     d[1] = s;
  722.     *d = 1;
  723.     } else
  724.         *d = 0;
  725.     return d;
  726. }
  727.  
  728.  
  729. long P_packset(s)                   /* convert s to a small-set */
  730. register long *s;
  731. {
  732.     if (*s++)
  733.         return *s;
  734.     else
  735.         return 0;
  736. }
  737.  
  738.  
  739.  
  740.  
  741.  
  742. /* Oregon Software Pascal extensions, courtesy of William Bader */
  743.  
  744. int P_getcmdline(l, h, line)
  745. int l, h;
  746. Char *line;
  747. {
  748.     int i, len;
  749.     char *s;
  750.     
  751.     h = h - l + 1;
  752.     len = 0;
  753.     for(i = 1; i < P_argc; i++) {
  754.     s = P_argv[i];
  755.     while (*s) {
  756.         if (len >= h) return len;
  757.         line[len++] = *s++;
  758.     }
  759.     if (len >= h) return len;
  760.     line[len++] = ' ';
  761.     }
  762.     return len;
  763. }
  764.  
  765. Void TimeStamp(Day, Month, Year, Hour, Min, Sec)
  766. int *Day, *Month, *Year, *Hour, *Min, *Sec;
  767. {
  768. #ifndef NO_TIME
  769.     struct tm *tm;
  770.     long clock;
  771.  
  772.     time(&clock);
  773.     tm = localtime(&clock);
  774.     *Day = tm->tm_mday;
  775.     *Month = tm->tm_mon + 1;        /* Jan = 0 */
  776.     *Year = tm->tm_year;
  777.     if (*Year < 1900)
  778.     *Year += 1900;     /* year since 1900 */
  779.     *Hour = tm->tm_hour;
  780.     *Min = tm->tm_min;
  781.     *Sec = tm->tm_sec;
  782. #endif
  783. }
  784.  
  785.  
  786.  
  787.  
  788. /* SUN Berkeley Pascal extensions */
  789.  
  790. Void P_sun_argv(s, len, n)
  791. register char *s;
  792. register int len, n;
  793. {
  794.     register char *cp;
  795.  
  796.     if ((unsigned)n < P_argc)
  797.     cp = P_argv[n];
  798.     else
  799.     cp = "";
  800.     while (*cp && --len >= 0)
  801.     *s++ = *cp++;
  802.     while (--len >= 0)
  803.     *s++ = ' ';
  804. }
  805.  
  806.  
  807.  
  808.  
  809. int _OutMem()
  810. {
  811.     return _Escape(-2);
  812. }
  813.  
  814. int _CaseCheck()
  815. {
  816.     return _Escape(-9);
  817. }
  818.  
  819. int _NilCheck()
  820. {
  821.     return _Escape(-3);
  822. }
  823.  
  824.  
  825.  
  826.  
  827.  
  828. /* The following is suitable for the HP Pascal operating system.
  829.    It might want to be revised when emulating another system. */
  830.  
  831. char *_ShowEscape(buf, code, ior, prefix)
  832. char *buf, *prefix;
  833. int code, ior;
  834. {
  835.     char *bufp;
  836.  
  837.     if (prefix && *prefix) {
  838.         strcpy(buf, prefix);
  839.     strcat(buf, ": ");
  840.         bufp = buf + strlen(buf);
  841.     } else {
  842.         bufp = buf;
  843.     }
  844.     if (code == -10) {
  845.         sprintf(bufp, "Pascal system I/O error %d", ior);
  846.         switch (ior) {
  847.             case 3:
  848.                 strcat(buf, " (illegal I/O request)");
  849.                 break;
  850.             case 7:
  851.                 strcat(buf, " (bad file name)");
  852.                 break;
  853.             case FileNotFound:   /*10*/
  854.                 strcat(buf, " (file not found)");
  855.                 break;
  856.             case FileNotOpen:    /*13*/
  857.                 strcat(buf, " (file not open)");
  858.                 break;
  859.             case BadInputFormat: /*14*/
  860.                 strcat(buf, " (bad input format)");
  861.                 break;
  862.             case 24:
  863.                 strcat(buf, " (not open for reading)");
  864.                 break;
  865.             case 25:
  866.                 strcat(buf, " (not open for writing)");
  867.                 break;
  868.             case 26:
  869.                 strcat(buf, " (not open for direct access)");
  870.                 break;
  871.             case 28:
  872.                 strcat(buf, " (string subscript out of range)");
  873.                 break;
  874.             case EndOfFile:      /*30*/
  875.                 strcat(buf, " (end-of-file)");
  876.                 break;
  877.             case FileWriteError: /*38*/
  878.         strcat(buf, " (file write error)");
  879.         break;
  880.         }
  881.     } else {
  882.         sprintf(bufp, "Pascal system error %d", code);
  883.         switch (code) {
  884.             case -2:
  885.                 strcat(buf, " (out of memory)");
  886.                 break;
  887.             case -3:
  888.                 strcat(buf, " (reference to NIL pointer)");
  889.                 break;
  890.             case -4:
  891.                 strcat(buf, " (integer overflow)");
  892.                 break;
  893.             case -5:
  894.                 strcat(buf, " (divide by zero)");
  895.                 break;
  896.             case -6:
  897.                 strcat(buf, " (real math overflow)");
  898.                 break;
  899.             case -8:
  900.                 strcat(buf, " (value range error)");
  901.                 break;
  902.             case -9:
  903.                 strcat(buf, " (CASE value range error)");
  904.                 break;
  905.             case -12:
  906.                 strcat(buf, " (bus error)");
  907.                 break;
  908.             case -20:
  909.                 strcat(buf, " (stopped by user)");
  910.                 break;
  911.         }
  912.     }
  913.     return buf;
  914. }
  915.  
  916.  
  917. int _Escape(code)
  918. int code;
  919. {
  920.     char buf[100];
  921.  
  922.     P_escapecode = code;
  923.     if (__top_jb) {
  924.     __p2c_jmp_buf *jb = __top_jb;
  925.     __top_jb = jb->next;
  926.     longjmp(jb->jbuf, 1);
  927.     }
  928.     if (code == 0)
  929.         exit(0);
  930.     if (code == -1)
  931.         exit(1);
  932.     fprintf(stderr, "%s\n", _ShowEscape(buf, P_escapecode, P_ioresult, ""));
  933.     exit(1);
  934. }
  935.  
  936. int _EscIO(code)
  937. int code;
  938. {
  939.     P_ioresult = code;
  940.     return _Escape(-10);
  941. }
  942.  
  943.  
  944.  
  945.  
  946. /* End. */
  947.  
  948.  
  949.  
  950.